home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / gximage2.c < prev    next >
C/C++ Source or Header  |  1997-03-02  |  14KB  |  471 lines

  1. /* Copyright (C) 1989, 1995, 1996, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gximage2.c */
  20. /* General monochrome image rendering */
  21. #include "gx.h"
  22. #include "memory_.h"
  23. #include "gpcheck.h"
  24. #include "gserrors.h"
  25. #include "gxfixed.h"
  26. #include "gxarith.h"
  27. #include "gxmatrix.h"
  28. #include "gsccolor.h"
  29. #include "gspaint.h"
  30. #include "gsutil.h"
  31. #include "gxdevice.h"
  32. #include "gxcmap.h"
  33. #include "gxdcolor.h"
  34. #include "gxistate.h"
  35. #include "gzpath.h"
  36. #include "gxdevmem.h"
  37. #include "gdevmem.h"            /* for mem_mono_device */
  38. #include "gxcpath.h"
  39. #include "gximage.h"
  40. #include "gzht.h"
  41.  
  42. /* ------ Strategy procedure ------ */
  43.  
  44. /* We can bypass X clipping for portrait monochrome images. */
  45. private irender_proc(image_render_mono);
  46. private irender_proc_t
  47. image_strategy_mono(gx_image_enum *penum)
  48. {    /* Use slow loop for imagemask with a halftone, */
  49.     /* or for a non-default logical operation. */
  50.     penum->slow_loop =
  51.       (penum->masked && !color_is_pure(&penum->icolor1)) ||
  52.       penum->use_rop;
  53.     if ( penum->spp == 1 )
  54.       { if ( !(penum->slow_loop || penum->posture != image_portrait) )
  55.           penum->clip_image &= ~(image_clip_xmin | image_clip_xmax);
  56.         if_debug0('b', "[b]render=mono\n");
  57.         /* Precompute values needed for rasterizing. */
  58.         penum->dxx =
  59.           float2fixed(penum->matrix.xx + fixed2float(fixed_epsilon) / 2);
  60.         return image_render_mono;
  61.       }
  62.     return 0;
  63. }
  64.  
  65. void
  66. gs_gximage2_init(gs_memory_t *mem)
  67. {    image_strategies.mono = image_strategy_mono;
  68. }
  69.  
  70. /* ------ Rendering procedure ------ */
  71.  
  72. /* Rendering procedure for the general case of displaying a */
  73. /* monochrome image, dealing with multiple bit-per-sample images, */
  74. /* general transformations, and arbitrary single-component */
  75. /* color spaces (DeviceGray, CIEBasedA, Separation, Indexed). */
  76. /* This procedure handles a single scan line. */
  77. private int
  78. image_render_mono(gx_image_enum *penum, const byte *buffer, int data_x,
  79.   uint w, int h, gx_device *dev)
  80. {    const gs_imager_state *pis = penum->pis;
  81.     gs_logical_operation_t lop = penum->log_op;
  82.     const int masked = penum->masked;
  83.     const gs_color_space *pcs;        /* only set for non-masks */
  84.     cs_proc_remap_color((*remap_color));    /* ditto */
  85.     gs_client_color cc;
  86.     const gx_color_map_procs *cmap_procs = gx_device_cmap_procs(dev);
  87.     cmap_proc_gray((*map_gray)) = cmap_procs->map_gray;
  88.     gx_device_color *pdevc = &penum->icolor1; /* color for masking */
  89.     /* Make sure the cache setup matches the graphics state. */
  90.     /* Also determine whether all tiles fit in the cache. */
  91.     int tiles_fit = gx_check_tile_cache(pis);
  92. #define image_set_gray(sample_value)\
  93.    { pdevc = &penum->clues[sample_value].dev_color;\
  94.      if ( !color_is_set(pdevc) )\
  95.       { if ( penum->device_color )\
  96.       (*map_gray)(byte2frac(sample_value), pdevc, pis, dev, gs_color_select_source);\
  97.     else\
  98.       { decode_sample(sample_value, cc, 0);\
  99.         (*remap_color)(&cc, pcs, pdevc, pis, dev, gs_color_select_source);\
  100.       }\
  101.       }\
  102.      else if ( !color_is_pure(pdevc) )\
  103.       { if ( !tiles_fit )\
  104.          { code = gx_color_load_select(pdevc, pis, dev, gs_color_select_source);\
  105.        if ( code < 0 ) return code;\
  106.      }\
  107.       }\
  108.    }
  109.     gx_dda_fixed_point next;    /* (y not used in fast loop) */
  110.     gx_dda_step_fixed dxx2, dxx3, dxx4; /* (not used in all loops) */
  111.     register const byte *psrc = buffer + data_x;
  112.     const byte *endp = psrc + w;
  113.     const byte *stop = endp;
  114.     fixed xrun;            /* x at start of run */
  115.     register byte run;        /* run value */
  116.     int htrun =            /* halftone run value */
  117.       (masked ? 255 : -2);
  118.     int code = 0;
  119.  
  120.     if ( h == 0 )
  121.       return 0;
  122.     next = penum->dda.pixel0;
  123.     xrun = dda_current(next.x);
  124.     if ( !masked )
  125.       { pcs = penum->pcs;        /* (may not be set for masks) */
  126.         remap_color = pcs->type->remap_color;
  127.       }
  128.     run = *psrc;
  129.     /* Find the last transition in the input. */
  130.     { byte last = stop[-1];
  131.       while ( stop > psrc && stop[-1] == last )
  132.         --stop;
  133.     }
  134.     if ( penum->slow_loop || penum->posture != image_portrait )
  135.       { /* Skewed, rotated, or imagemask with a halftone. */
  136.         fixed yrun;
  137.         const fixed pdyx = dda_current(penum->dda.row.x) - penum->cur.x;
  138.         const fixed pdyy = dda_current(penum->dda.row.y) - penum->cur.y;
  139.         dev_proc_fill_parallelogram((*fill_pgram)) =
  140.           dev_proc(dev, fill_parallelogram);
  141.  
  142. #define xl dda_current(next.x)
  143. #define ytf dda_current(next.y)
  144.         yrun = ytf;
  145.         if ( masked )
  146.           { pdevc = &penum->icolor1;
  147.             code = gx_color_load(pdevc, pis, dev);
  148.             if ( code < 0 )
  149.           return code;
  150.         if ( stop <= psrc )
  151.           goto last;
  152.         if ( penum->posture == image_portrait )
  153.           { /* We don't have to worry about the Y DDA, */
  154.             /* and the fill regions are rectangles. */
  155.             /* Calculate multiples of the DDA step. */
  156.             dxx2 = next.x.step;
  157.             dda_step_add(dxx2, next.x.step);
  158.             dxx3 = dxx2;
  159.             dda_step_add(dxx3, next.x.step);
  160.             dxx4 = dxx3;
  161.             dda_step_add(dxx4, next.x.step);
  162.             for ( ; ; )
  163.               { /* Skip a run of zeros. */
  164.             while ( !psrc[0] )
  165.               if ( !psrc[1] )
  166.                 { if ( !psrc[2] )
  167.                     { if ( !psrc[3] )
  168.                     { psrc += 4;
  169.                       dda_state_next(next.x.state, dxx4);
  170.                       continue;
  171.                     }
  172.                   psrc += 3;
  173.                   dda_state_next(next.x.state, dxx3);
  174.                   break;
  175.                 }
  176.                   psrc += 2;
  177.                   dda_state_next(next.x.state, dxx2);
  178.                   break;
  179.                 }
  180.               else
  181.                 { ++psrc;
  182.                   dda_next(next.x);
  183.                   break;
  184.                 }
  185.                 xrun = xl;
  186.             if ( psrc >= stop )
  187.               break;
  188.             for ( ; *psrc; ++psrc )
  189.               dda_next(next.x);
  190.             code = (*fill_pgram)(dev, xrun, yrun, xl - xrun,
  191.                          fixed_0, fixed_0, pdyy,
  192.                          pdevc, lop);
  193.             if ( code < 0 )
  194.               return code;
  195.             if ( psrc >= stop )
  196.               break;
  197.               }
  198.           }
  199.         else
  200.          for ( ; ; )
  201.           { for ( ; !*psrc; ++psrc )
  202.               { dda_next(next.x);
  203.                 dda_next(next.y);
  204.               }
  205.             yrun = ytf;
  206.             xrun = xl;
  207.             if ( psrc >= stop )
  208.               break;
  209.             for ( ; *psrc; ++psrc )
  210.               { dda_next(next.x);
  211.             dda_next(next.y);
  212.                }
  213.             code = (*fill_pgram)(dev, xrun, yrun, xl - xrun,
  214.                      ytf - yrun, pdyx, pdyy, pdevc, lop);
  215.             if ( code < 0 )
  216.               return code;
  217.             if ( psrc >= stop )
  218.               break;
  219.           }
  220.           }
  221.         else if ( penum->posture == image_portrait ||
  222.               penum->posture == image_landscape
  223.             )
  224.           {        /* Not masked, and we can fill runs quickly. */
  225.          if ( stop <= psrc )
  226.            goto last;
  227.          for ( ; ; )
  228.           { if ( *psrc != run )
  229.               { if ( run != htrun )
  230.               { htrun = run;
  231.                 image_set_gray(run);
  232.               }
  233.                 code = (*fill_pgram)(dev, xrun, yrun, xl - xrun,
  234.                          ytf - yrun, pdyx, pdyy,
  235.                          pdevc, lop);
  236.             if ( code < 0 )
  237.               return code;
  238.                 yrun = ytf;
  239.                 xrun = xl;
  240.                 if ( psrc >= stop )
  241.               break;
  242.             run = *psrc;
  243.               }
  244.             psrc++;
  245.             dda_next(next.x);
  246.             dda_next(next.y);
  247.           }
  248.           }
  249.         else        /* not masked */
  250.           { /* Since we have to check for the end after every pixel */
  251.         /* anyway, we may as well avoid the last-run code. */
  252.         stop = endp;
  253.         for ( ; ; )
  254.           { /* We can't skip large constant regions quickly, */
  255.             /* because this leads to rounding errors. */
  256.             /* Just fill the region between xrun and xl. */
  257.             psrc++;
  258.             if ( run != htrun )
  259.               { htrun = run;
  260.                 image_set_gray(run);
  261.               }
  262.             code = (*fill_pgram)(dev, xrun, yrun, xl - xrun,
  263.                      ytf - yrun, pdyx, pdyy, pdevc, lop);
  264.             if ( code < 0 )
  265.               return code;
  266.             yrun = ytf;
  267.             xrun = xl;
  268.             if ( psrc > stop )
  269.               { --psrc;
  270.                 break;
  271.               }
  272.             run = psrc[-1];
  273.             dda_next(next.x);
  274.             dda_next(next.y);        /* harmless if no skew */
  275.           }
  276.           }
  277.         /* Fill the last run. */
  278. last:        if ( stop < endp && (*stop || !masked) )
  279.           { if ( !masked )
  280.           { image_set_gray(*stop);
  281.           }
  282.             dda_advance(next.x, endp - stop);
  283.             dda_advance(next.y, endp - stop);
  284.         code = (*fill_pgram)(dev, xrun, yrun, xl - xrun,
  285.                      ytf - yrun, pdyx, pdyy, pdevc, lop);
  286.           }
  287. #undef xl
  288. #undef ytf
  289.       }
  290.     else            /* fast loop */
  291.       { /* No skew, and not imagemask with a halftone. */
  292.         const fixed adjust = penum->adjust;
  293.         const fixed dxx = penum->dxx;
  294.         fixed xa = (dxx >= 0 ? adjust : -adjust);
  295.         const int yt = penum->yci, iht = penum->hci;
  296.         dev_proc_fill_rectangle((*fill_proc)) =
  297.           dev_proc(dev, fill_rectangle);
  298.         dev_proc_strip_tile_rectangle((*tile_proc)) =
  299.           dev_proc(dev, strip_tile_rectangle);
  300.         dev_proc_copy_mono((*copy_mono_proc)) =
  301.           dev_proc(dev, copy_mono);
  302.         /*
  303.          * If each pixel is likely to fit in a single halftone tile,
  304.          * determine that now (tile_offset = offset of row within tile).
  305.          * Don't do this for band devices; they handle halftone fills
  306.          * more efficiently than copy_mono.
  307.          */
  308.         int bstart;
  309.         int phase_x;
  310.         int tile_offset =
  311.           ((*dev_proc(dev, get_band))(dev, yt, &bstart) == 0 ?
  312.            gx_check_tile_size(pis,
  313.                   fixed2int_ceiling(any_abs(dxx) + (xa << 1)),
  314.                   yt, iht, gs_color_select_source, &phase_x) :
  315.            -1);
  316.         int xmin = fixed2int_pixround(penum->clip_outer.p.x);
  317.         int xmax = fixed2int_pixround(penum->clip_outer.q.x);
  318.  
  319. #define xl dda_current(next.x)
  320.         /* Fold the adjustment into xrun and xl, */
  321.         /* including the +0.5-epsilon for rounding. */
  322.         xrun = xrun - xa + (fixed_half - fixed_epsilon);
  323.         dda_translate(next.x, xa + (fixed_half - fixed_epsilon));
  324.         xa <<= 1;
  325.         /* Calculate multiples of the DDA step. */
  326.         dxx2 = next.x.step;
  327.         dda_step_add(dxx2, next.x.step);
  328.         dxx3 = dxx2;
  329.         dda_step_add(dxx3, next.x.step);
  330.         dxx4 = dxx3;
  331.         dda_step_add(dxx4, next.x.step);
  332.         if ( stop > psrc )
  333.          for ( ; ; )
  334.           {    /* Skip large constant regions quickly, */
  335.         /* but don't slow down transitions too much. */
  336. skf:        if ( psrc[0] == run )
  337.           { if ( psrc[1] == run )
  338.               { if ( psrc[2] == run )
  339.               { if ( psrc[3] == run )
  340.                   { psrc += 4;
  341.                 dda_state_next(next.x.state, dxx4);
  342.                 goto skf;
  343.                   }
  344.                 else
  345.                   { psrc += 4;
  346.                 dda_state_next(next.x.state, dxx3);
  347.                   }
  348.               }
  349.                 else
  350.               { psrc += 3;
  351.                 dda_state_next(next.x.state, dxx2);
  352.               }
  353.               }
  354.             else
  355.               { psrc += 2;
  356.             dda_next(next.x);
  357.               }
  358.           }
  359.         else
  360.           psrc++;
  361.         { /* Now fill the region between xrun and xl. */
  362.           int xi = fixed2int_var(xrun);
  363.           int wi = fixed2int_var(xl) - xi;
  364.           int xei, tsx;
  365.           const gx_strip_bitmap *tile;
  366.  
  367.           if ( wi <= 0 )
  368.             { if ( wi == 0 ) goto mt;
  369.               xi += wi, wi = -wi;
  370.             }
  371.           if ( (xei = xi + wi) > xmax || xi < xmin )
  372.             { /* Do X clipping */
  373.               if ( xi < xmin )
  374.             wi -= xmin - xi, xi = xmin;
  375.               if ( xei > xmax )
  376.             wi -= xei - xmax;
  377.               if ( wi <= 0 )
  378.             goto mt;
  379.             }
  380.           switch ( run )
  381.             {
  382.             case 0:
  383.               if ( masked ) goto mt;
  384.               if ( !color_is_pure(&penum->icolor0) ) goto ht;
  385.               code = (*fill_proc)(dev, xi, yt, wi, iht,
  386.                       penum->icolor0.colors.pure);
  387.               break;
  388.             case 255:        /* just for speed */
  389.               if ( !color_is_pure(&penum->icolor1) ) goto ht;
  390.               code = (*fill_proc)(dev, xi, yt, wi, iht,
  391.                       penum->icolor1.colors.pure);
  392.               break;
  393.             default:
  394. ht:              /* Use halftone if needed */
  395.               if ( run != htrun )
  396.             { image_set_gray(run);
  397.               htrun = run;
  398.             }
  399.               /* We open-code gx_fill_rectangle, */
  400.               /* because we've done some of the work for */
  401.               /* halftone tiles in advance. */
  402.               if ( color_is_pure(pdevc) )
  403.             { code = (*fill_proc)(dev, xi, yt, wi, iht,
  404.                           pdevc->colors.pure);
  405.             }
  406.               else if ( !color_is_binary_halftone(pdevc) )
  407.             { code =
  408.                 gx_fill_rectangle_device_rop(xi, yt, wi, iht,
  409.                              pdevc, dev, lop);
  410.             }
  411.               else if ( tile_offset >= 0 &&
  412.                     (tile = &pdevc->colors.binary.b_tile->tiles,
  413.                  (tsx = (xi + phase_x) % tile->rep_width) + wi <= tile->size.x)
  414.                   )
  415.             { /* The pixel(s) fit(s) in a single (binary) tile. */
  416.               byte *row = tile->data + tile_offset;
  417.               code = (*copy_mono_proc)
  418.                 (dev, row, tsx, tile->raster, gx_no_bitmap_id,
  419.                  xi, yt, wi, iht,
  420.                  pdevc->colors.binary.color[0],
  421.                  pdevc->colors.binary.color[1]);
  422.             }
  423.               else
  424.             { code = (*tile_proc)(dev,
  425.                           &pdevc->colors.binary.b_tile->tiles,
  426.                           xi, yt, wi, iht,
  427.                           pdevc->colors.binary.color[0],
  428.                           pdevc->colors.binary.color[1],
  429.                           pdevc->phase.x, pdevc->phase.y);
  430.             }
  431.             }
  432.             if ( code < 0 ) return code;
  433. mt:            xrun = xl - xa;    /* original xa << 1 */
  434.             if ( psrc > stop )
  435.               { --psrc;
  436.                 break;
  437.               }
  438.             run = psrc[-1];
  439.         }
  440.         dda_next(next.x);
  441.           }
  442.         /* Fill the last run. */
  443.         if ( *stop != 0 || !masked )
  444.           { int xi = fixed2int_var(xrun);
  445.             int wi, xei;
  446.  
  447.         dda_advance(next.x, endp - stop);
  448.         wi = fixed2int_var(xl) - xi;
  449.         if ( wi <= 0 )
  450.           { if ( wi == 0 ) goto lmt;
  451.             xi += wi, wi = -wi;
  452.           }
  453.         if ( (xei = xi + wi) > xmax || xi < xmin )
  454.           { /* Do X clipping */
  455.             if ( xi < xmin )
  456.               wi -= xmin - xi, xi = xmin;
  457.             if ( xei > xmax )
  458.               wi -= xei - xmax;
  459.             if ( wi <= 0 )
  460.               goto lmt;
  461.           }
  462.         image_set_gray(*stop);
  463.         code = gx_fill_rectangle_device_rop(xi, yt, wi, iht,
  464.                             pdevc, dev, lop);
  465. lmt:            ;
  466.           }
  467.         }
  468. #undef xl
  469.     return (code < 0 ? code : 1);
  470. }
  471.